Tabs

Posted on 2023-04-23 by

henrikvilhelmberglund

He're were going to make a Tabs component for displaying tabs. I already made a simple tab system myself for displaying the code but it could probably be improved.

It can be good to think about how the user is going to use our component instead of just starting blindly.

<script>
	// to hide errors
	let Tabs;
	let Tab;
	let isLoggedIn;
	let username;
</script>

<!-- 

  <Tabs>
    <Tab title="tab 1">Hey hey</Tab>
    {#if isLoggedIn}
		<Tab title="secret tab 2">Logged in content 1</Tab>
		<Tab title="secret tab 3">Logged in content 2</Tab>
    {/if}
    <Tab title={username}>info for {username}</Tab>
  </Tabs>
-->
<!-- alternative way, but we think just having an if statement to show or hide them feels better -->

<!-- <Tab shouldShow={isLoggedIn}>Tab content</Tab> -->
<style>
</style>

This looks fine! Let's start building it.

<script>
	import Tabs from "./Tabs.svelte";
	import Tab from "./Tab.svelte";
	let isLoggedIn;
	let username = "Henrik";
	let selectedTab = "2";
</script>

<div>
	<input type="checkbox" bind:checked={isLoggedIn} />
	<input type="text" bind:value={username} />
</div>

<button on:click={() => (selectedTab = "4")}>Change tab</button>

<Tabs bind:selectedTab>
	<Tab id="1" title="tab 1">Hey hey</Tab>
	{#if isLoggedIn}
		<Tab id="2" title="secret tab 2">Logged in content 1</Tab>
		<Tab id="3" title="secret tab 3">Logged in content 2</Tab>
	{/if}
	<Tab id="4" title={username}>info for {username}</Tab>
</Tabs>
<style>
</style>

At first the login tabs ended up at the end even though they had IDs of 2 and 3 .

I solved it by using an object instead which works but from my understanding it is not guaranteed to work since objects are not ordered.

Let's try fixing the array implementation instead:

Instead of pushing the value directly we sort them by their ID when assigning using the spread operator. As a bonus since we're assigning we don't need the ugly titles = titles either. Credits to Believe Lody for this solution!
<script>
	import { setContext } from "svelte";
	import { writable } from "svelte/store";
	// we want to be able to view one tab at a time
	export let selectedTab = "1";
	let selectedTabStore = writable(selectedTab);
	$: $selectedTabStore = selectedTab;
	// for making the bind: work
	$: updateProps($selectedTabStore);

	function updateProps(value) {
		selectedTab = value;
	}

	setContext("selectedTab", selectedTabStore);

	let titles = [];
	setContext("tabTitles", {
		registerTab(id, title) {
			// titles.push({ id, title });
			titles = [...titles, { id, title }].sort((a, b) => a.id - b.id);
		},
		updateTitle(id, title) {
			const tabIndex = titles.findIndex((title) => title.id === id);
			titles[tabIndex].title = title;
		},
		unregisterTab(id) {
			const tabIndex = titles.findIndex((title) => title.id === id);
			if (tabIndex > -1) {
				titles.splice(tabIndex, 1);
				titles = titles;
			}
		},
	});
</script>

<div>
	{#each titles as { id, title }}
		<!-- not reactive yet because context is not reactive across components, we need a store as well -->
		<button class:selected={$selectedTabStore === id} on:click={() => ($selectedTabStore = id)}
			>{title}</button>
	{/each}
</div>

<slot />

<style>
	button.selected {
		background: black;
		color: white;
	}
</style>

There we go! In this example we used slots, context, reactive context using a store, bind and reactive statements to end up with a fairly simple store.